JUC-Callable&ThreadPool

本文详细介绍了Java中线程池的使用方法,包括固定大小线程池、缓存线程池、单个线程的线程池及可定时执行任务的线程池。同时,深入探讨了Callable接口与FutureTask的运用,展示如何让线程执行任务并获取返回结果。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

引入:

创建执行线程的方式有四种:

1.继承Thread类

2.实现Runnable接口

3.实现Callable接口

4.线程池

前两种用的最多,今天就来演示一下后两种方法创建线程,即实现Callable接口和使用线程池。

1、实现Callable接口:

Callable是JUC中的一个接口,相较于实现Runnable接口的方式,其方法可以有返回值,并且可以抛出异常。另外,使用Callable接口需要FutureTask实现类的支持,用于接收运算结果。FutureTask是Future接口的实现类。

实际操作:

package com.tongtong.app7;

import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;

public class TestCallable {

    public static void main(String[] args) {

        ThreadDemo td = new ThreadDemo();

        //1.使用Callable,需要FutureTask实现类的支持,用于接收运算结果
        FutureTask<Integer> result = new FutureTask<>(td);

        new Thread(result).start();

        //2.接收线程运算后的结果
        try {
            Integer sum = result.get();
            System.out.println(sum);
            System.out.println("---------------------");
        } catch (InterruptedException | ExecutionException e) {
            e.printStackTrace();
        }
    }
}

class ThreadDemo implements Callable<Integer>{

    @Override
    public Integer call() throws Exception {

        int sum = 0;

        for(int i = 0;i <= 100000; i++){
            sum += i;
        }

        return sum;
    }
}

 

2、线程池

线程池:提供一个线程队列,队列中保存着所有等待状态的线程。避免了创建与销毁额外开销,提高了响应的速度。

二、线程池的体系结构:

java.util.concurrent.Executor:负责线程的使用与调度的根接口
    |--ExecutorService 子接口:线程池的主要接口(重要)
       |--ThreadPoolExecutor 线程池的实现类
       |--ScheduledExecutorService:子接口:负责线程的调度
           |--ScheduledThreadPoolExecutor :继承ThreadPoolExecutor,实现ScheduledExecutorService

三、工具类 : Executors

ExecutorService newFixedThreadPool() : 创建固定大小的线程池
ExecutorService newCachedThreadPool() : 缓存线程池,线程池的数量不固定,可以根据需求自动的更改数量
ExecutorService newSingleThreadExecutor() : 创建单个线程池,线程池中只有一个线程
ScheduledExecutorService newScheduledThreadPool() : 创建固定大小的线程,可以延迟或定时的执行任务

实际操作: 

1、创建固定大小的线程池(ExecutorService newFixedThreadPool())

package com.tongtong.app8;

import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.*;

public class TestThreadPool {

    public static void main(String[] args) throws ExecutionException, InterruptedException {

        //1、创建固定大小的线程池
        ExecutorService pool = Executors.newFixedThreadPool(5);

        List<Future<Integer>> list = new ArrayList<>();

        for(int i = 0;i < 10; i++){ //分配10个任务
            //接收返回值
            Future<Integer> future = pool.submit(new Callable<Integer>() {

                @Override
                public Integer call() throws Exception {

                    int sum = 0;

                    for(int i =0;i <= 100;i++){
                        sum += i;
                    }
                    return sum;
                }
            });

            list.add(future);
        }
        //关闭线程池
        pool.shutdown();

        for(Future<Integer> future : list){
            System.out.println(future.get());
        }

    }
}

 

2、创建缓存线程池(ExecutorService newCachedThreadPool())

package com.tongtong.app8;

import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.*;

public class TestThreadPool {

    public static void main(String[] args) throws ExecutionException, InterruptedException {

        //1、创建缓存的线程池
        ExecutorService pool = Executors.newCachedThreadPool(5);

        List<Future<Integer>> list = new ArrayList<>();

        for(int i = 0;i < 10; i++){ //分配10个任务
            //接收返回值
            Future<Integer> future = pool.submit(new Callable<Integer>() {

                @Override
                public Integer call() throws Exception {

                    int sum = 0;

                    for(int i =0;i <= 100;i++){
                        sum += i;
                    }
                    return sum;
                }
            });

            list.add(future);
        }
        //关闭线程池
        pool.shutdown();

        for(Future<Integer> future : list){
            System.out.println(future.get());
        }

    }
}

3、创建单个线程的线程池(ExecutorService newSingleThreadExecutor())

package com.tongtong.app8;

import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.*;

public class TestThreadPool {

    public static void main(String[] args) throws ExecutionException, InterruptedException {

        //1、创建线程池
        ExecutorService pool = Executors.newSingleThreadExecutor();

        List<Future<Integer>> list = new ArrayList<>();

        for(int i = 0;i < 10; i++){
            Future<Integer> future = pool.submit(new Callable<Integer>() {

                @Override
                public Integer call() throws Exception {

                    int sum = 0;

                    for(int i =0;i <= 100;i++){
                        sum += i;
                    }
                    return sum;
                }
            });

            list.add(future);
        }

        pool.shutdown();

        for(Future<Integer> future : list){
            System.out.println(future.get());
        }
    }
}

4、创建固定大小的线程池,可以延迟或定时的执行任务(ScheduledExecutorService newScheduledThreadPool())

package com.tongtong.app8;

import java.util.Random;
import java.util.concurrent.*;

public class TestScheduledThreadPool {

    public static void main(String[] args) throws ExecutionException, InterruptedException {

        ScheduledExecutorService pool = Executors.newScheduledThreadPool(5);
        for(int i = 0;i < 5; i++){
            Future<Integer> result = pool.schedule(new Callable<Integer>() {

                @Override
                public Integer call() throws Exception {
                    int num = new Random().nextInt(100);
                    System.out.println(Thread.currentThread().getName() + " : " + num);
                    return num;
                }
            },1, TimeUnit.SECONDS); //定时执行

            System.out.println(result.get());
        }

        pool.shutdown();
    }
}

 

 

 

 

### Java JUC AQS 并发编程 抽象队列同步器 使用教程 源码解析 #### 什么是AQS? `AbstractQueuedSynchronizer`(简称AQS),作为Java并发包中的核心组件之一,提供了用于实现锁和其他同步器的基础框架。它不仅简化了锁和同步工具的创建过程,还提高了这些工具的工作效率[^2]。 #### 类图结构与工作原理 AQS的设计围绕着一个FIFO(先进先出)等待队列展开,该队列由多个节点组成,每个节点代表了一个正在等待获取资源的线程。每当有新的竞争者未能立即获得所需资源时,就会被构造成一个新的节点并加入到这个队列之中;而当现有持有者释放其持有的资源之后,则会从队头开始依次唤醒后续等待者去尝试占有资源[^5]。 #### 同步模式分类 为了适应不同场景下的需求,AQS支持两种主要类型的同步方式——独占式以及共享式: - **独占式**:一次只允许单个线程访问临界区,在这种情况下其他任何试图进入同一区域内的请求都将被迫挂起直到前序操作完成为止; - **共享式**:允许多个读取者同时存在而不互相干扰,只要不存在写入动作发生即可保持一致性和安全性[^3]。 #### 自定义同步器的关键接口 对于想要利用AQS来构建特定行为逻辑的新类型而言,开发者通常需要重载以下几个抽象方法以适配具体的应用环境: - `tryAcquire(int arg)` 和 `tryRelease(int arg)` - `tryAcquireShared(int arg)` 及 `tryReleaseShared(int arg)` - `isHeldExclusively()` 上述函数分别对应于独占/共享模式下对资源的操作控制流程,通过合理地覆盖它们可以轻松打造出满足业务特性要求的各种高级别同步原语[^1]。 ```java public class CustomSync extends AbstractQueuedSynchronizer { protected boolean tryAcquire(int acquires) { // 实现具体的独占式获取逻辑 return super.tryAcquire(acquires); } protected boolean tryRelease(int releases) { // 实现具体的独占式释放逻辑 return super.tryRelease(releases); } } ``` #### 队列管理机制详解 在实际运行过程中,AQS内部维护了一条双向链表形式的数据结构用来存储各个待处理的任务单元。每当新成员到来之时便会调用`enqueue()`方法将其追加至末端位置上形成完整的链条关系网状链接,并且借助CAS指令保证整个插入过程的安全可靠性质不受外界因素影响破坏[^4]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值